home *** CD-ROM | disk | FTP | other *** search
/ CU Amiga Super CD-ROM 15 / CU Amiga Magazine's Super CD-ROM 15 (1997)(EMAP Images)(GB)[!][issue 1997-10].iso / CUCD / Graphics / Ghostscript / source / gdevx.c < prev    next >
C/C++ Source or Header  |  1997-05-04  |  38KB  |  1,251 lines

  1. /* Copyright (C) 1989, 1995, 1996, 1997 Aladdin Enterprises.  All rights reserved.
  2.   
  3.   This file is part of Aladdin Ghostscript.
  4.   
  5.   Aladdin Ghostscript is distributed with NO WARRANTY OF ANY KIND.  No author
  6.   or distributor accepts any responsibility for the consequences of using it,
  7.   or for whether it serves any particular purpose or works at all, unless he
  8.   or she says so in writing.  Refer to the Aladdin Ghostscript Free Public
  9.   License (the "License") for full details.
  10.   
  11.   Every copy of Aladdin Ghostscript must include a copy of the License,
  12.   normally in a plain ASCII text file named PUBLIC.  The License grants you
  13.   the right to copy, modify and redistribute Aladdin Ghostscript, but only
  14.   under certain conditions described in the License.  Among other things, the
  15.   License requires that the copyright notice and this notice be preserved on
  16.   all copies.
  17. */
  18.  
  19. /* gdevx.c */
  20. /* X Windows driver for Ghostscript library */
  21. /* The X include files include <sys/types.h>, which, on some machines */
  22. /* at least, define uint, ushort, and ulong, which std.h also defines. */
  23. /* std.h has taken care of this. */
  24. #include "gx.h"            /* for gx_bitmap; includes std.h */
  25. #include "math_.h"
  26. #include "memory_.h"
  27. #include "x_.h"
  28. #include "gserrors.h"
  29. #include "gsparam.h"
  30. #include "gxdevice.h"
  31. #include "gdevx.h"
  32.  
  33. /* Define whether to update after every write, for debugging. */
  34. bool always_update = false;
  35.  
  36. /* Define the maximum size of the temporary pixmap for copy_mono */
  37. /* that we are willing to leave lying around in the server */
  38. /* between uses.  (Assume 32-bit ints here!) */
  39. private int max_temp_pixmap = 20000;
  40.  
  41. /* Forward references */
  42. private int set_tile(P2(gx_device *, const gx_strip_bitmap *));
  43. private void free_cp(P1(gx_device *));
  44. /* Screen updating machinery */
  45. #define update_init(dev)\
  46.   ((gx_device_X *)(dev))->up_area = 0,\
  47.   ((gx_device_X *)(dev))->up_count = 0
  48. #define update_flush(dev)\
  49.   if ( ((gx_device_X *)(dev))->up_area != 0 ) update_do_flush(dev)
  50. private void update_do_flush(P1(gx_device *));
  51. private void x_send_event(P2(gx_device *, Atom));
  52.  
  53. /* Procedures */
  54.  
  55. extern int gdev_x_open(P1(gx_device_X *));
  56. private dev_proc_open_device(x_open);
  57. private dev_proc_get_initial_matrix(x_get_initial_matrix);
  58. private dev_proc_sync_output(x_sync);
  59. private dev_proc_output_page(x_output_page);
  60. private dev_proc_close_device(x_close);
  61. private dev_proc_map_rgb_color(x_map_rgb_color);
  62. private dev_proc_map_color_rgb(x_map_color_rgb);
  63. private dev_proc_fill_rectangle(x_fill_rectangle);
  64. private dev_proc_copy_mono(x_copy_mono);
  65. private dev_proc_copy_color(x_copy_color);
  66. private dev_proc_get_bits(x_get_bits);
  67. private dev_proc_put_params(x_put_params);
  68. dev_proc_get_xfont_procs(x_get_xfont_procs);
  69. private dev_proc_get_page_device(x_get_page_device);
  70. private dev_proc_strip_tile_rectangle(x_strip_tile_rectangle);
  71.  
  72. /* The device descriptor */
  73. private gx_device_procs x_procs = {
  74.     x_open,
  75.     x_get_initial_matrix,
  76.     x_sync,
  77.     x_output_page,
  78.     x_close,
  79.     x_map_rgb_color,
  80.     x_map_color_rgb,
  81.     x_fill_rectangle,
  82.     NULL,            /* tile_rectangle */
  83.     x_copy_mono,
  84.     x_copy_color,
  85.     NULL,            /* draw_line */
  86.     x_get_bits,
  87.     NULL,            /* get_params */
  88.     x_put_params,
  89.     NULL,            /* map_cmyk_color */
  90.     x_get_xfont_procs,
  91.     NULL,            /* get_xfont_device */
  92.     NULL,            /* map_rgb_alpha_color */
  93.     x_get_page_device,
  94.     NULL,            /* get_alpha_bits */
  95.     NULL,            /* copy_alpha */
  96.     NULL,            /* get_band */
  97.     NULL,            /* copy_rop */
  98.     NULL,            /* fill_path */
  99.     NULL,            /* stroke_path */
  100.     NULL,            /* fill_mask */
  101.     NULL,            /* fill_trapezoid */
  102.     NULL,            /* fill_parallelogram */
  103.     NULL,            /* fill_triangle */
  104.     NULL,            /* draw_thin_line */
  105.     NULL,            /* begin_image */
  106.     NULL,            /* image_data */
  107.     NULL,            /* end_image */
  108.     x_strip_tile_rectangle
  109. };
  110.  
  111. /* The instance is public. */
  112. gx_device_X far_data gs_x11_device = {
  113.     std_device_color_body(gx_device_X, &x_procs, "x11",
  114.       FAKE_RES*DEFAULT_WIDTH_10THS/10, FAKE_RES*DEFAULT_HEIGHT_10THS/10, /* x and y extent (nominal) */
  115.       FAKE_RES, FAKE_RES,    /* x and y density (nominal) */
  116.       /*dci_color(*/24, 255, 256/*)*/),
  117.     { 0 },            /* std_procs */
  118.     1 /*true*/,        /* IsPageDevice */
  119.     { /* image */
  120.       0, 0,            /* width, height */
  121.       0, XYBitmap, NULL,    /* xoffset, format, data */
  122.       LSBFirst, 8,        /* byte-order, bitmap-unit */
  123.       MSBFirst, 8, 1,    /* bitmap-bit-order, bitmap-pad, depth */
  124.       0, 1,            /* bytes_per_line, bits_per_pixel */
  125.       0, 0, 0,        /* red_mask, green_mask, blue_mask */
  126.       NULL,            /* *obdata */
  127.        { NULL,            /* *(*create_image)() */
  128.          NULL,            /* (*destroy_image)() */
  129.          NULL,            /* (*get_pixel)() */
  130.          NULL,            /* (*put_pixel)() */
  131.          NULL,            /* *(*sub_image)() */
  132.          NULL            /* (*add_pixel)() */
  133.        },
  134.     },
  135.     NULL, NULL,        /* dpy, scr */
  136.                 /* (connection not initialized) */
  137.     NULL,            /* vinfo */
  138.     (Colormap)None,        /* cmap */
  139.     (Window)None,        /* win */
  140.     NULL,            /* gc */
  141.     (Window)None,        /* pwin */
  142.     (Pixmap)0,        /* bpixmap */
  143.     0,            /* ghostview */
  144.     (Window)None,        /* mwin */
  145. #if HaveStdCMap
  146.     NULL,            /* std_cmap */
  147. #endif
  148.     { identity_matrix_body },    /* initial matrix (filled in) */
  149.     (Atom)0, (Atom)0, (Atom)0,    /* Atoms: NEXT, PAGE, DONE */
  150.      { 0, 0, 0, 0 }, 0, 0,    /* update, up_area, up_count */
  151.     (Pixmap)0,        /* dest */
  152.     0L, (ulong)~0L,        /* colors_or, colors_and */
  153.      { /* cp */
  154.        (Pixmap)0,        /* pixmap */
  155.        NULL,        /* gc */
  156.        -1, -1        /* raster, height */
  157.      },
  158.      { /* ht */
  159.        (Pixmap)None,        /* pixmap */
  160.        (Pixmap)None,        /* no_pixmap */
  161.        gx_no_bitmap_id,        /* id */
  162.        0, 0, 0,            /* width, height, raster */
  163.        0, 0                /* fore_c, back_c */
  164.      },
  165.     GXcopy,            /* function */
  166.     FillSolid,        /* fill_style */
  167.     0,            /* font */
  168.     0, 0,            /* back_color, fore_color */
  169.     0, 0,            /* background, foreground */
  170.     NULL,            /* dither_colors */
  171.     0, 0,            /* color_mask, num_rgb */
  172.     NULL, 0,        /* dynamic_colors, max_dynamic_colors */
  173.     0, 0,            /* dynamic_size, dynamic_allocs */
  174.     0, 0,            /* borderColor, borderWidth */
  175.     NULL,            /* geometry */
  176.     128, 5,            /* maxGrayRamp, maxRGBRamp */
  177.     NULL,            /* palette */
  178.     NULL, NULL, NULL,    /* regularFonts, symbolFonts, dingbatFonts */
  179.     NULL, NULL, NULL,    /* regular_fonts, symbol_fonts, dingbat_fonts */
  180.     1, 1,            /* useXFonts, useFontExtensions */
  181.     1, 0,            /* useScalableFonts, logXFonts */
  182.     0.0, 0.0,        /* xResolution, yResolution */
  183.     1,            /* useBackingPixmap */
  184.     1, 1,            /* useXPutImage, useXSetTile */
  185. };
  186.  
  187. /* If XPutImage doesn't work, do it ourselves. */
  188. private int alt_put_image(P11(gx_device *dev, Display *dpy, Drawable win,
  189.   GC gc, XImage *pi, int sx, int sy, int dx, int dy, unsigned w, unsigned h));
  190. #define put_image(dpy,win,gc,im,sx,sy,x,y,w,h)\
  191.   do {\
  192.     if ( xdev->useXPutImage ) {\
  193.       XPutImage(dpy,win,gc,im,sx,sy,x,y,w,h);\
  194.     } else {\
  195.       int code_ = alt_put_image(dev,dpy,win,gc,im,sx,sy,x,y,w,h);\
  196.       if ( code_ < 0 ) return code_;\
  197.     }\
  198.   } while (0)
  199.  
  200. /* Open the device.  Most of the code is in gdevxini.c. */
  201. private int
  202. x_open(gx_device *dev)
  203. {
  204.     gx_device_X *xdev = (gx_device_X *)dev;
  205.     int code = gdev_x_open(xdev);
  206.  
  207.     if (code < 0) return code;
  208.     update_init(dev);
  209.     return 0;
  210. }
  211.  
  212. /* Close the device. */
  213. private int
  214. x_close(gx_device *dev)
  215. {
  216.     gx_device_X *xdev = (gx_device_X *)dev;
  217.  
  218.     if (xdev->ghostview) x_send_event(dev, xdev->DONE);
  219.     if (xdev->vinfo) {
  220.     XFree((char *)xdev->vinfo);
  221.     xdev->vinfo = NULL;
  222.     }
  223.     if (xdev->dither_colors) {
  224.     if (gx_device_has_color(xdev))
  225. #define cube(r) (r*r*r)
  226.         gs_free((char *)xdev->dither_colors, sizeof(x_pixel),
  227.             cube(xdev->color_info.dither_colors), "x11_rgb_cube");
  228. #undef cube
  229.     else
  230.         gs_free((char *)xdev->dither_colors, sizeof(x_pixel),
  231.             xdev->color_info.dither_grays, "x11_gray_ramp");
  232.     xdev->dither_colors = NULL;
  233.     }
  234.     if (xdev->dynamic_colors) {
  235.     int i;
  236.     for (i = 0; i < xdev->dynamic_size; i++) {
  237.         x11color *xcp = (*xdev->dynamic_colors)[i];
  238.         x11color *next;
  239.         while (xcp) {
  240.         next = xcp->next;
  241.         gs_free((char *)xcp, sizeof(x11color), 1, "x11_dynamic_color");
  242.         xcp = next;
  243.         }
  244.     }
  245.     gs_free((char *)xdev->dynamic_colors, sizeof(x11color *),
  246.         xdev->dynamic_size, "x11_dynamic_colors");
  247.     xdev->dynamic_colors = NULL;
  248.     }
  249.     while (xdev->regular_fonts) {
  250.     x11fontmap *font = xdev->regular_fonts;
  251.     xdev->regular_fonts = font->next;
  252.     if (font->std_names) XFreeFontNames(font->std_names);
  253.     if (font->iso_names) XFreeFontNames(font->iso_names);
  254.     gs_free(font->x11_name, sizeof(char), strlen(font->x11_name)+1,
  255.         "x11_font_x11name");
  256.     gs_free(font->ps_name, sizeof(char), strlen(font->ps_name)+1,
  257.         "x11_font_psname");
  258.     gs_free((char *)font, sizeof(x11fontmap), 1, "x11_fontmap");
  259.     }
  260.     while (xdev->symbol_fonts) {
  261.     x11fontmap *font = xdev->symbol_fonts;
  262.     xdev->symbol_fonts = font->next;
  263.     if (font->std_names) XFreeFontNames(font->std_names);
  264.     if (font->iso_names) XFreeFontNames(font->iso_names);
  265.     gs_free(font->x11_name, sizeof(char), strlen(font->x11_name)+1,
  266.         "x11_font_x11name");
  267.     gs_free(font->ps_name, sizeof(char), strlen(font->ps_name)+1,
  268.         "x11_font_psname");
  269.     gs_free((char *)font, sizeof(x11fontmap), 1, "x11_fontmap");
  270.     }
  271.     while (xdev->dingbat_fonts) {
  272.     x11fontmap *font = xdev->dingbat_fonts;
  273.     xdev->dingbat_fonts = font->next;
  274.     if (font->std_names) XFreeFontNames(font->std_names);
  275.     if (font->iso_names) XFreeFontNames(font->iso_names);
  276.     gs_free(font->x11_name, sizeof(char), strlen(font->x11_name)+1,
  277.         "x11_font_x11name");
  278.     gs_free(font->ps_name, sizeof(char), strlen(font->ps_name)+1,
  279.         "x11_font_psname");
  280.     gs_free((char *)font, sizeof(x11fontmap), 1, "x11_fontmap");
  281.     }
  282.     XCloseDisplay(xdev->dpy);
  283.     return 0;
  284. }
  285.  
  286. /* Define a table for computing N * X_max_color_value / D for 0 <= N <= D, */
  287. /* 1 <= D <= 7. */
  288. /* This requires a multiply and a divide otherwise; */
  289. /* integer multiply and divide are slow on all platforms. */
  290. #define cv_fraction(n, d) ((ushort)(X_max_color_value * (n) / (d)))
  291. #define nd(n, d) cv_fraction(n, d)
  292. private const ushort cv_tab1[] = {
  293.   nd(0,1), nd(1,1)
  294. };
  295. private const ushort cv_tab2[] = {
  296.   nd(0,2), nd(1,2), nd(2,2)
  297. };
  298. private const ushort cv_tab3[] = {
  299.   nd(0,3), nd(1,3), nd(2,3), nd(3,3)
  300. };
  301. private const ushort cv_tab4[] = {
  302.   nd(0,4), nd(1,4), nd(2,4), nd(3,4), nd(4,4)
  303. };
  304. private const ushort cv_tab5[] = {
  305.   nd(0,5), nd(1,5), nd(2,5), nd(3,5), nd(4,5), nd(5,5)
  306. };
  307. private const ushort cv_tab6[] = {
  308.   nd(0,6), nd(1,6), nd(2,6), nd(3,6), nd(4,6), nd(5,6), nd(6,6)
  309. };
  310. private const ushort cv_tab7[] = {
  311.   nd(0,7), nd(1,7), nd(2,7), nd(3,7), nd(4,7), nd(5,7), nd(6,7), nd(7,7)
  312. };
  313. private const ushort _ds *cv_tables[] = {
  314.   0, cv_tab1, cv_tab2, cv_tab3, cv_tab4, cv_tab5, cv_tab6, cv_tab7
  315. };
  316.  
  317. /* Map a color.  The "device colors" are just r,g,b packed together. */
  318. private gx_color_index
  319. x_map_rgb_color(register gx_device *dev,
  320.         gx_color_value r, gx_color_value g, gx_color_value b)
  321. {
  322.     gx_device_X *xdev = (gx_device_X *)dev;
  323.     /* X and ghostscript both use shorts for color values */
  324.     unsigned short dr = r & xdev->color_mask;    /* Nearest color that */
  325.     unsigned short dg = g & xdev->color_mask;    /* the X device can   */
  326.     unsigned short db = b & xdev->color_mask;    /* represent          */
  327.     unsigned short cv_max = X_max_color_value & xdev->color_mask;
  328.     int cv_denom = (gx_max_color_value + 1);
  329.  
  330.     /* Foreground and background get special treatment: */
  331.     /* They may be mapped to other colors. */
  332.     if ( (dr | dg | db) == 0 ) {    /* i.e., all 0 */
  333.     return xdev->foreground;
  334.     }
  335.     if ( (dr & dg & db) == cv_max ) {    /* i.e., all max value */
  336.     return xdev->background;
  337.     }
  338. #if HaveStdCMap
  339.     /* check the standard colormap first */
  340.     if (xdev->std_cmap) {
  341.     XStandardColormap *cmap = xdev->std_cmap;
  342.  
  343.     if (gx_device_has_color(xdev)) {
  344.         unsigned short cr, cg, cb;        /* rgb cube indices */
  345.         unsigned short cvr, cvg, cvb;    /* color value on cube */
  346.  
  347.         cr = r * (cmap->red_max + 1) / cv_denom;
  348.         cg = g * (cmap->green_max + 1) / cv_denom;
  349.         cb = b * (cmap->blue_max + 1) / cv_denom;
  350.         cvr = X_max_color_value * cr / cmap->red_max;
  351.         cvg = X_max_color_value * cg / cmap->green_max;
  352.         cvb = X_max_color_value * cb / cmap->blue_max;
  353.         if ((abs((int)r - (int)cvr) & xdev->color_mask) == 0 &&
  354.         (abs((int)g - (int)cvg) & xdev->color_mask) == 0 &&
  355.         (abs((int)b - (int)cvb) & xdev->color_mask) == 0)
  356.         return cr * cmap->red_mult + cg * cmap->green_mult +
  357.                cb * cmap->blue_mult + cmap->base_pixel;
  358.     } else {
  359.         unsigned short cr;
  360.         unsigned short cvr;
  361.         int dither_grays = xdev->color_info.dither_grays;
  362.  
  363.         cr = r * dither_grays / cv_denom;
  364.         cvr = X_max_color_value * cr / cmap->red_max;
  365.         if ((abs((int)r - (int)cvr) & xdev->color_mask) == 0)
  366.         return cr * cmap->red_mult + cmap->base_pixel;
  367.     }
  368.     } else
  369. #endif
  370.     /* If there is no standard colormap, check the dither cube/ramp */
  371.     if (xdev->dither_colors) {
  372.     if (gx_device_has_color(xdev)) {
  373.         unsigned short cr, cg, cb;        /* rgb cube indices */
  374.         unsigned short cvr, cvg, cvb;    /* color value on cube */
  375.         int dither_rgb = xdev->color_info.dither_colors;
  376.         unsigned short max_rgb = dither_rgb - 1;
  377.  
  378.         cr = r * dither_rgb / cv_denom;
  379.         cg = g * dither_rgb / cv_denom;
  380.         cb = b * dither_rgb / cv_denom;
  381.         if ( max_rgb < countof(cv_tables) )
  382.           { const ushort *cv_tab = cv_tables[max_rgb];
  383.         cvr = cv_tab[cr];
  384.         cvg = cv_tab[cg];
  385.         cvb = cv_tab[cb];
  386.           }
  387.         else
  388.           { cvr = cv_fraction(cr, max_rgb);
  389.         cvg = cv_fraction(cg, max_rgb);
  390.         cvb = cv_fraction(cb, max_rgb);
  391.           }
  392.         if ((abs((int)r - (int)cvr) & xdev->color_mask) == 0 &&
  393.         (abs((int)g - (int)cvg) & xdev->color_mask) == 0 &&
  394.         (abs((int)b - (int)cvb) & xdev->color_mask) == 0) {
  395.         return xdev->dither_colors[cube_index(cr, cg, cb)];
  396.         }
  397.     } else {
  398.         unsigned short cr;
  399.         unsigned short cvr;
  400.         int dither_grays = xdev->color_info.dither_grays;
  401.         unsigned short max_gray = dither_grays - 1;
  402.  
  403.         cr = r * dither_grays / cv_denom;
  404.         cvr = (X_max_color_value * cr / max_gray);
  405.         if ((abs((int)r - (int)cvr) & xdev->color_mask) == 0)
  406.         return xdev->dither_colors[cr];
  407.     }
  408.     }
  409.     /* Finally look through the list of dynamic colors */
  410.     if (xdev->dynamic_colors) {
  411.     int i = (dr ^ dg ^ db) >> (16 - xdev->vinfo->bits_per_rgb);
  412.     x11color *xcp = (*xdev->dynamic_colors)[i];
  413.     x11color *last = NULL;
  414.     XColor xc;
  415.  
  416.     while (xcp) {
  417.         if (xcp->color.red == dr && xcp->color.green == dg &&
  418.         xcp->color.blue == db) {
  419.         if (last) {
  420.             last->next = xcp->next;
  421.             xcp->next = (*xdev->dynamic_colors)[i];
  422.             (*xdev->dynamic_colors)[i] = xcp;
  423.         }
  424.         if (xcp->color.pad) return xcp->color.pixel;
  425.         else return gx_no_color_index;
  426.         }
  427.         last = xcp;
  428.         xcp = xcp->next;
  429.     }
  430.  
  431.     /* If not in our list of dynamic colors, */
  432.     /* ask the X server and add an entry. */
  433.     /* First check if dynamic table is exhausted */
  434.     if (xdev->dynamic_allocs > xdev->max_dynamic_colors)
  435.         return gx_no_color_index;
  436.     xcp = (x11color *) gs_malloc(sizeof(x11color), 1, "x11_dynamic_color");
  437.     if (!xcp) return gx_no_color_index;
  438.     xc.red   = xcp->color.red   = dr;
  439.     xc.green = xcp->color.green = dg;
  440.     xc.blue  = xcp->color.blue  = db;
  441.     xcp->next = (*xdev->dynamic_colors)[i];
  442.     (*xdev->dynamic_colors)[i] = xcp;
  443.     xdev->dynamic_allocs++;
  444.     if (XAllocColor(xdev->dpy, xdev->cmap, &xc)) {
  445.         xcp->color.pixel = xc.pixel;
  446.         xcp->color.pad = True;
  447.         return xc.pixel;
  448.     } else {
  449.         xcp->color.pad = False;
  450.         return gx_no_color_index;
  451.     }
  452.     }
  453.     return gx_no_color_index;
  454. }
  455.  
  456.  
  457. /* Map a "device color" back to r-g-b. */
  458. /* This doesn't happen often, so we just ask the display */
  459. /* Foreground and background may be mapped to other colors, so */
  460. /* they are handled specially. */
  461. private int
  462. x_map_color_rgb(register gx_device *dev, gx_color_index color,
  463.         gx_color_value prgb[3])
  464. {
  465.     gx_device_X *xdev = (gx_device_X *)dev;
  466.  
  467.     if (color == xdev->foreground)
  468.     prgb[0] = prgb[1] = prgb[2] = 0;
  469.     else if (color == xdev->background)
  470.     prgb[0] = prgb[1] = prgb[2] = gx_max_color_value;
  471.     else {
  472.     XColor xc;
  473.  
  474.     xc.pixel = color;
  475.     XQueryColor(xdev->dpy, xdev->cmap, &xc);
  476.     prgb[0] = xc.red;
  477.     prgb[1] = xc.green;
  478.     prgb[2] = xc.blue;
  479.     }
  480.     return 0;
  481. }
  482.  
  483. /* Get initial matrix for X device. */
  484. /* This conflicts seriously with the code for page devices; */
  485. /* we only do it if Ghostview is active. */
  486. private void
  487. x_get_initial_matrix(register gx_device *dev, register gs_matrix *pmat)
  488. {
  489.     gx_device_X *xdev = (gx_device_X *)dev;
  490.  
  491.     if ( !xdev->ghostview )
  492.       { gx_default_get_initial_matrix(dev, pmat);
  493.     return;
  494.       }
  495.     pmat->xx = xdev->initial_matrix.xx;
  496.     pmat->xy = xdev->initial_matrix.xy;
  497.     pmat->yx = xdev->initial_matrix.yx;
  498.     pmat->yy = xdev->initial_matrix.yy;
  499.     pmat->tx = xdev->initial_matrix.tx;
  500.     pmat->ty = xdev->initial_matrix.ty;
  501. }
  502.  
  503. /* Synchronize the display with the commands already given */
  504. private int
  505. x_sync(register gx_device *dev)
  506. {
  507.     gx_device_X *xdev = (gx_device_X *)dev;
  508.  
  509.     update_flush(dev);
  510.     XFlush(xdev->dpy);
  511.     return 0;
  512. }
  513.  
  514. /* Send event to ghostview process */
  515. private void
  516. x_send_event(gx_device *dev, Atom msg)
  517. {
  518.     gx_device_X *xdev = (gx_device_X *)dev;
  519.     XEvent event;
  520.  
  521.     event.xclient.type = ClientMessage;
  522.     event.xclient.display = xdev->dpy;
  523.     event.xclient.window = xdev->win;
  524.     event.xclient.message_type = msg;
  525.     event.xclient.format = 32;
  526.     event.xclient.data.l[0] = xdev->mwin;
  527.     event.xclient.data.l[1] = xdev->dest;
  528.     XSendEvent(xdev->dpy, xdev->win, False, 0, &event);
  529. }
  530.  
  531. /* Output "page" */
  532. private int
  533. x_output_page(gx_device *dev, int num_copies, int flush)
  534. {
  535.     gx_device_X *xdev = (gx_device_X *)dev;
  536.  
  537.     x_sync(dev);
  538.  
  539.     /* Send ghostview a "page" client event */
  540.     /* Wait for a "next" client event */
  541.     if (xdev->ghostview) {
  542.     XEvent event;
  543.  
  544.     x_send_event(dev, xdev->PAGE);
  545.     XNextEvent(xdev->dpy, &event);
  546.     while (event.type != ClientMessage ||
  547.            event.xclient.message_type != xdev->NEXT) {
  548.         XNextEvent(xdev->dpy, &event);
  549.     }
  550.     }
  551.     return 0;
  552. }
  553.  
  554. /* Fill a rectangle with a color. */
  555. private int
  556. x_fill_rectangle(register gx_device *dev,
  557.          int x, int y, int w, int h, gx_color_index color)
  558. {
  559.     gx_device_X *xdev = (gx_device_X *)dev;
  560.  
  561.     fit_fill(dev, x, y, w, h);
  562.     set_fill_style(FillSolid);
  563.     set_fore_color(color);
  564.     set_function(GXcopy);
  565.     XFillRectangle(xdev->dpy, xdev->dest, xdev->gc, x, y, w, h);
  566.     /* If we are filling the entire screen, reset */
  567.     /* colors_or and colors_and.  It's wasteful to do this */
  568.     /* on every operation, but there's no separate driver routine */
  569.     /* for erasepage (yet). */
  570.     if (x == 0 && y == 0 && w == xdev->width && h == xdev->height) {
  571.     if (color == xdev->foreground || color == xdev->background) {
  572.         if (xdev->dynamic_colors) {
  573.         int i;
  574.         for (i = 0; i < xdev->dynamic_size; i++) {
  575.             x11color *xcp = (*xdev->dynamic_colors)[i];
  576.             x11color *next;
  577.             while (xcp) {
  578.             next = xcp->next;
  579.             if (xcp->color.pad) {
  580.                 XFreeColors(xdev->dpy, xdev->cmap,
  581.                     &xcp->color.pixel, 1, 0);
  582.             }
  583.             gs_free((char *)xcp, sizeof(x11color), 1,
  584.                 "x11_dynamic_color");
  585.             xcp = next;
  586.             }
  587.             (*xdev->dynamic_colors)[i] = NULL;
  588.         }
  589.         xdev->dynamic_allocs = 0;
  590.         }
  591.     }
  592.     xdev->colors_or = xdev->colors_and = color;
  593.     }
  594.     if (xdev->bpixmap != (Pixmap) 0) {
  595.     x_update_add(dev, x, y, w, h);
  596.     }
  597. #ifdef DEBUG
  598.     if (gs_debug['F'])
  599.     dprintf5("[F] fill (%d,%d):(%d,%d) %ld\n",
  600.          x, y, w, h, (long)color);
  601. #endif
  602.     return 0;
  603. }
  604.  
  605. /* Copy a monochrome bitmap. */
  606. private int
  607. x_copy_mono(register gx_device *dev,
  608.         const byte *base, int sourcex, int raster, gx_bitmap_id id,
  609.         int x, int y, int w, int h,
  610.         gx_color_index zero, gx_color_index one)
  611. /*
  612.  * X doesn't directly support the simple operation of writing a color
  613.  * through a mask specified by an image.  The plot is the following:
  614.  *  If neither color is gx_no_color_index ("transparent"),
  615.  *    use XPutImage with the "copy" function as usual.
  616.  *  If the color either bitwise-includes or is bitwise-included-in
  617.  *    every color written to date
  618.  *    (a special optimization for writing black/white on color displays),
  619.  *    use XPutImage with an appropriate Boolean function.
  620.  *  Otherwise, do the following complicated stuff:
  621.  *    Create pixmap of depth 1 if necessary.
  622.  *    If foreground color is "transparent" then
  623.  *      invert the raster data.
  624.  *    Use XPutImage to copy the raster image to the newly
  625.  *      created Pixmap.
  626.  *    Install the Pixmap as the clip_mask in the X GC and
  627.  *      tweak the clip origin.
  628.  *    Do an XFillRectangle, fill style=solid, specifying a
  629.  *      rectangle the same size as the original raster data.
  630.  *    De-install the clip_mask.
  631.  */
  632. {
  633.     gx_device_X *xdev = (gx_device_X *)dev;
  634.     int function = GXcopy;
  635.  
  636.     x_pixel
  637.     bc = zero,
  638.     fc = one;
  639.  
  640.     fit_copy(dev, base, sourcex, raster, id, x, y, w, h);
  641.  
  642.     xdev->image.width = sourcex + w;
  643.     xdev->image.height = h;
  644.     xdev->image.data = (char *)base;
  645.     xdev->image.bytes_per_line = raster;
  646.     set_fill_style(FillSolid);
  647.  
  648.     /* Check for null, easy 1-color, hard 1-color, and 2-color cases. */
  649.     if (zero != gx_no_color_index) {
  650.     if (one != gx_no_color_index) {
  651.         /* 2-color case. */
  652.         /* Simply replace existing bits with what's in the image. */
  653.     } else if (!(~xdev->colors_and & bc)) {
  654.         function = GXand;
  655.         fc = ~(x_pixel) 0;
  656.     } else if (!(~bc & xdev->colors_or)) {
  657.         function = GXor;
  658.         fc = 0;
  659.     } else {
  660.         goto hard;
  661.     }
  662.     } else {
  663.     if (one == gx_no_color_index) {    /* no-op */
  664.         return 0;
  665.     } else if (!(~xdev->colors_and & fc)) {
  666.         function = GXand;
  667.         bc = ~(x_pixel) 0;
  668.     } else if (!(~fc & xdev->colors_or)) {
  669.         function = GXor;
  670.         bc = 0;
  671.     } else {
  672.         goto hard;
  673.     }
  674.     }
  675.     xdev->image.format = XYBitmap;
  676.     set_function(function);
  677.     if (bc != xdev->back_color) {
  678.     XSetBackground(xdev->dpy, xdev->gc, (xdev->back_color = bc));
  679.     }
  680.     if (fc != xdev->fore_color) {
  681.     XSetForeground(xdev->dpy, xdev->gc, (xdev->fore_color = fc));
  682.     }
  683.     if (zero != gx_no_color_index)
  684.     note_color(zero);
  685.     if (one != gx_no_color_index)
  686.     note_color(one);
  687.     put_image(xdev->dpy, xdev->dest, xdev->gc, &xdev->image,
  688.           sourcex, 0, x, y, w, h);
  689.  
  690.     goto out;
  691.  
  692. hard:    /* Handle the hard 1-color case. */
  693.     if (raster > xdev->cp.raster || h > xdev->cp.height) {
  694.     /* Must allocate a new pixmap and GC. */
  695.     /* Release the old ones first. */
  696.     free_cp(dev);
  697.  
  698.     /* Create the clipping pixmap, depth must be 1. */
  699.     xdev->cp.pixmap =
  700.         XCreatePixmap(xdev->dpy, xdev->win, raster << 3, h, 1);
  701.     if (xdev->cp.pixmap == (Pixmap) 0) {
  702.         lprintf("x_copy_mono: can't allocate pixmap\n");
  703.         return_error(gs_error_VMerror);
  704.     }
  705.     xdev->cp.gc = XCreateGC(xdev->dpy, xdev->cp.pixmap, 0, 0);
  706.     if (xdev->cp.gc == (GC) 0) {
  707.         lprintf("x_copy_mono: can't allocate GC\n");
  708.         return_error(gs_error_VMerror);
  709.     }
  710.     xdev->cp.raster = raster;
  711.     xdev->cp.height = h;
  712.     }
  713.     /* Initialize static mask image params */
  714.     xdev->image.format = XYBitmap;
  715.     set_function(GXcopy);
  716.  
  717.     /* Select polarity based on fg/bg transparency. */
  718.     if (one == gx_no_color_index) {    /* invert */
  719.     XSetBackground(xdev->dpy, xdev->cp.gc, (x_pixel) 1);
  720.     XSetForeground(xdev->dpy, xdev->cp.gc, (x_pixel) 0);
  721.     set_fore_color(zero);
  722.     } else {
  723.     XSetBackground(xdev->dpy, xdev->cp.gc, (x_pixel) 0);
  724.     XSetForeground(xdev->dpy, xdev->cp.gc, (x_pixel) 1);
  725.     set_fore_color(one);
  726.     }
  727.     put_image(xdev->dpy, xdev->cp.pixmap, xdev->cp.gc,
  728.           &xdev->image, sourcex, 0, 0, 0, w, h);
  729.  
  730.     /* Install as clipmask. */
  731.     XSetClipMask(xdev->dpy, xdev->gc, xdev->cp.pixmap);
  732.     XSetClipOrigin(xdev->dpy, xdev->gc, x, y);
  733.  
  734.     /*
  735.      * Draw a solid rectangle through the raster clip mask.
  736.      * Note fill style is guaranteed to be solid from above.
  737.      */
  738.     XFillRectangle(xdev->dpy, xdev->dest, xdev->gc, x, y, w, h);
  739.  
  740.     /* Tidy up.  Free the pixmap if it's big. */
  741.     XSetClipMask(xdev->dpy, xdev->gc, None);
  742.     if (raster * h > max_temp_pixmap)
  743.     free_cp(dev);
  744.  
  745. out:if (xdev->bpixmap != (Pixmap) 0) {
  746.     /* We wrote to the pixmap, so update the display now. */
  747.     x_update_add(dev, x, y, w, h);
  748.     }
  749.     return 0;
  750. }
  751.  
  752. /* Internal routine to free the GC and pixmap used for copying. */
  753. private void
  754. free_cp(register gx_device *dev)
  755. {
  756.     gx_device_X *xdev = (gx_device_X *)dev;
  757.  
  758.     if (xdev->cp.gc != NULL) {
  759.     XFreeGC(xdev->dpy, xdev->cp.gc);
  760.     xdev->cp.gc = NULL;
  761.     }
  762.     if (xdev->cp.pixmap != (Pixmap) 0) {
  763.     XFreePixmap(xdev->dpy, xdev->cp.pixmap);
  764.     xdev->cp.pixmap = (Pixmap) 0;
  765.     }
  766.     xdev->cp.raster = -1;    /* mark as unallocated */
  767. }
  768.  
  769. /* Copy a color bitmap. */
  770. private int
  771. x_copy_color(register gx_device *dev,
  772.          const byte *base, int sourcex, int raster, gx_bitmap_id id,
  773.          int x, int y, int w, int h)
  774. {    gx_device_X *xdev = (gx_device_X *)dev;
  775.     int depth = dev->color_info.depth;
  776.  
  777.     fit_copy(dev, base, sourcex, raster, id, x, y, w, h);
  778.     set_fill_style(FillSolid);
  779.     set_function(GXcopy);
  780.  
  781.     /* Filling with a colored halftone often gives rise to */
  782.     /* copy_color calls for a single pixel.  Check for this now. */
  783.  
  784.     if ( h == 1 && w == 1 )
  785.       {    uint sbit = sourcex * depth;
  786.         const byte *ptr = base + (sbit >> 3);
  787.         x_pixel pixel;
  788.         if ( depth < 8 )
  789.           pixel = (byte)(*ptr << (sbit & 7)) >> (8 - depth);
  790.         else
  791.           { pixel = *ptr++;
  792.             while ( (depth -= 8) > 0 )
  793.               pixel = (pixel << 8) + *ptr++;
  794.           }
  795.         set_fore_color(pixel);
  796.         XDrawPoint(xdev->dpy, xdev->dest, xdev->gc, x, y);
  797.       }
  798.     else
  799.       {    xdev->image.width = sourcex + w;
  800.         xdev->image.height = h;
  801.         xdev->image.format = ZPixmap;
  802.         xdev->image.data = (char *)base;
  803.         xdev->image.depth = depth;
  804.         xdev->image.bytes_per_line = raster;
  805.         xdev->image.bits_per_pixel = depth;
  806.         XPutImage(xdev->dpy, xdev->dest, xdev->gc, &xdev->image,
  807.               sourcex, 0, x, y, w, h);
  808.         xdev->image.depth = xdev->image.bits_per_pixel = 1;
  809.       }
  810.     if (xdev->bpixmap != (Pixmap) 0)
  811.       x_update_add(dev, x, y, w, h);
  812. #ifdef DEBUG
  813.     if (gs_debug['F'])
  814.       dprintf4("[F] copy_color (%d,%d):(%d,%d)\n",
  815.            x, y, w, h);
  816. #endif
  817.     return 0;
  818. }
  819.  
  820. private int
  821. x_get_bits(gx_device *dev, int y, byte *str, byte **actual_data)
  822. {    gx_device_X *xdev = (gx_device_X *)dev;
  823.     int depth = dev->color_info.depth;
  824.     uint width_bytes = gx_device_raster(dev, false);
  825.     XImage *image;
  826.  
  827.     /* Make sure the frame buffer is up to date. */
  828.     if ( xdev->up_area != 0 &&
  829.          y >= xdev->update.yo && y < xdev->update.ye
  830.        )
  831.       update_do_flush(dev);
  832.     /*
  833.      * The X library doesn't provide any way to specify the desired
  834.      * bit or byte ordering for the result, so we just hope for the best
  835.      * (big-endian).
  836.      */
  837.     image = XGetImage(xdev->dpy, xdev->dest, 0, y, dev->width, 1,
  838.               (1L << depth) - 1, ZPixmap);
  839.     memcpy(str, image->data, width_bytes);
  840.     XDestroyImage(image);
  841.     *actual_data = str;
  842.     return 0;
  843. }
  844.  
  845. /* Set the device parameters.  We reimplement this so we can resize */
  846. /* the window and avoid closing and reopening the device, and to add */
  847. /* .IsPageDevice. */
  848. private int
  849. x_put_params(gx_device *dev, gs_param_list *plist)
  850. {    gx_device_X *xdev = (gx_device_X *)dev;
  851.     bool is_open = dev->is_open;
  852.     int width = dev->width;
  853.     int height = dev->height;
  854.     float xres = dev->HWResolution[0];
  855.     float yres = dev->HWResolution[1];
  856.     long pwin = (long)xdev->pwin;
  857.     bool is_page = xdev->IsPageDevice;
  858.     bool save_is_page = xdev->IsPageDevice;
  859.     int ecode = 0, code;
  860.  
  861.     /* Handle extra parameters */
  862.     switch ( code = param_read_long(plist, "WindowID", &pwin) )
  863.     {
  864.     case 0:
  865.     case 1:
  866.         break;
  867.     default:
  868.         ecode = code;
  869.         param_signal_error(plist, "GSVIEW", ecode);
  870.     }
  871.  
  872.     switch ( code = param_read_bool(plist, ".IsPageDevice", &is_page) )
  873.       {
  874.       case 0:
  875.       case 1:
  876.         break;
  877.       default:
  878.         ecode = code;
  879.         param_signal_error(plist, "GSVIEW", ecode);
  880.       }
  881.  
  882.     if ( ecode < 0 )
  883.       return ecode;
  884.  
  885.     /* Unless we specified a new window ID, */
  886.     /* prevent gx_default_put_params from closing the device. */
  887.     if ( pwin == (long)xdev->pwin )
  888.       dev->is_open = false;
  889.     xdev->IsPageDevice = is_page;
  890.     code = gx_default_put_params(dev, plist);
  891.     dev->is_open = is_open;
  892.     if ( code < 0 )
  893.       {    /* Undo setting of .IsPageDevice */
  894.         xdev->IsPageDevice = save_is_page;
  895.         return code;
  896.       }
  897.  
  898.     if ( pwin != (long)xdev->pwin )
  899.       {    if ( xdev->is_open )
  900.           gs_closedevice(dev);
  901.         xdev->pwin = (Window)pwin;
  902.       }
  903.  
  904.     /* If the device is open, resize the window. */
  905.     /* Don't do this if Ghostview is active. */
  906.     if ( xdev->is_open && !xdev->ghostview &&
  907.           (dev->width != width || dev->height != height ||
  908.            dev->HWResolution[0] != xres || dev->HWResolution[1] != yres)
  909.        )
  910.       {    int dw = dev->width - width;
  911.         int dh = dev->height - height;
  912.         double qx = dev->HWResolution[0] / xres;
  913.         double qy = dev->HWResolution[1] / xres;
  914.         if ( dw != 0 || dh != 0 )
  915.           {    XResizeWindow(xdev->dpy, xdev->win,
  916.                       dev->width, dev->height);
  917.             if (xdev->bpixmap != (Pixmap) 0)
  918.               {    XFreePixmap(xdev->dpy, xdev->bpixmap);
  919.                 xdev->bpixmap = (Pixmap) 0;
  920.               }
  921.             xdev->dest = 0;
  922.             gdev_x_clear_window(xdev);
  923.           }
  924.         /* Attempt to update the initial matrix in a sensible way. */
  925.         /* The whole handling of the initial matrix is a hack! */
  926.         if ( xdev->initial_matrix.xy == 0 )
  927.           {    if ( xdev->initial_matrix.xx < 0 )
  928.               {    /* 180 degree rotation */
  929.                 xdev->initial_matrix.tx += dw;
  930.               }
  931.             else
  932.               {    /* no rotation */
  933.                 xdev->initial_matrix.ty += dh;
  934.               }
  935.           }
  936.         else
  937.           {    if ( xdev->initial_matrix.xy < 0 )
  938.               {    /* 90 degree rotation */
  939.                 xdev->initial_matrix.tx += dh;
  940.                 xdev->initial_matrix.ty += dw;
  941.               }
  942.             else
  943.               {    /* 270 degree rotation */
  944.               }
  945.           }
  946.         xdev->initial_matrix.xx *= qx;
  947.         xdev->initial_matrix.xy *= qx;
  948.         xdev->initial_matrix.yx *= qy;
  949.         xdev->initial_matrix.yy *= qy;
  950.       }
  951.  
  952.     return 0;
  953. }
  954.  
  955. /* Get the page device.  We reimplement this so that we can make this */
  956. /* device be a page device conditionally. */
  957. private gx_device *
  958. x_get_page_device(gx_device *dev)
  959. {    return (((gx_device_X *)dev)->IsPageDevice ? dev : (gx_device *)0);
  960. }
  961.  
  962. /* Tile a rectangle. */
  963. private int
  964. x_strip_tile_rectangle(register gx_device *dev, const gx_strip_bitmap *tiles,
  965.                int x, int y, int w, int h,
  966.                gx_color_index zero, gx_color_index one,
  967.                int px, int py)
  968. {
  969.     gx_device_X *xdev = (gx_device_X *)dev;
  970.  
  971.     /* Give up if one color is transparent, or if the tile is colored. */
  972.     /* We should implement the latter someday, since X can handle it. */
  973.  
  974.     if (one == gx_no_color_index || zero == gx_no_color_index)
  975.       return gx_default_strip_tile_rectangle(dev, tiles, x, y, w, h,
  976.                          zero, one, px, py);
  977.  
  978.     /* For the moment, give up if the phase or shift is non-zero. */
  979.     if (tiles->shift | px | py)
  980.       return gx_default_strip_tile_rectangle(dev, tiles, x, y, w, h,
  981.                          zero, one, px, py);
  982.  
  983.     fit_fill(dev, x, y, w, h);
  984.  
  985.     /* Imaging with a halftone often gives rise to very small */
  986.     /* tile_rectangle calls.  Check for this now. */
  987.  
  988.     if ( h <= 2 && w <= 2 )
  989.       {    int j;
  990.     set_fill_style(FillSolid);
  991.     set_function(GXcopy);
  992.     for ( j = 0; j < h; j++ )
  993.       { const byte *ptr =
  994.           tiles->data + ((y + j) % tiles->rep_height) * tiles->raster;
  995.         int i;
  996.         for ( i = 0; i < w; i++ )
  997.           {    uint tx = (x + i) % tiles->rep_width;
  998.         byte mask = 0x80 >> (tx & 7);
  999.         x_pixel pixel = (ptr[tx >> 3] & mask ? one : zero);
  1000.         set_fore_color(pixel);
  1001.         XDrawPoint(xdev->dpy, xdev->dest, xdev->gc,
  1002.                x + i, y + j);
  1003.           }
  1004.       }
  1005.     if (xdev->bpixmap != (Pixmap) 0) {
  1006.       x_update_add(dev, x, y, w, h);
  1007.     }
  1008.     return 0;
  1009.       }
  1010.  
  1011.     /*
  1012.      * Remember, an X tile is already filled with particular
  1013.      * pixel values (i.e., colors).  Therefore if we are changing
  1014.      * fore/background color, we must invalidate the tile (using
  1015.      * the same technique as in set_tile).  This problem only
  1016.      * bites when using grayscale -- you may want to change
  1017.      * fg/bg but use the same halftone screen.
  1018.      */
  1019.     if ((zero != xdev->ht.back_c) || (one != xdev->ht.fore_c))
  1020.     xdev->ht.id = ~tiles->id;    /* force reload */
  1021.  
  1022.     set_back_color(zero);
  1023.     set_fore_color(one);
  1024.     if (!set_tile(dev, tiles))
  1025.       {    /* Bad news.  Fall back to the default algorithm. */
  1026.     return gx_default_strip_tile_rectangle(dev, tiles, x, y, w, h,
  1027.                            zero, one, px, py);
  1028.       }
  1029.     /* Use the tile to fill the rectangle */
  1030.     set_fill_style(FillTiled);
  1031.     set_function(GXcopy);
  1032.     XFillRectangle(xdev->dpy, xdev->dest, xdev->gc, x, y, w, h);
  1033.     if (xdev->bpixmap != (Pixmap) 0) {
  1034.       x_update_add(dev, x, y, w, h);
  1035.     }
  1036. #ifdef DEBUG
  1037.     if (gs_debug['F'])
  1038.     dprintf6("[F] tile (%d,%d):(%d,%d) %ld,%ld\n",
  1039.          x, y, w, h, (long)zero, (long)one);
  1040. #endif
  1041.     return 0;
  1042. }
  1043.  
  1044. /* Set up with a specified tile. */
  1045. /* Return false if we can't do it for some reason. */
  1046. private int
  1047. set_tile(register gx_device *dev, register const gx_strip_bitmap *tile)
  1048. {
  1049.     gx_device_X *xdev = (gx_device_X *)dev;
  1050.  
  1051. #ifdef DEBUG
  1052.     if (gs_debug['T'])
  1053.     return 0;
  1054. #endif
  1055.     if (tile->id == xdev->ht.id && tile->id != gx_no_bitmap_id)
  1056.     return xdev->useXSetTile;
  1057.     /* Set up the tile Pixmap */
  1058.     if (tile->size.x != xdev->ht.width ||
  1059.     tile->size.y != xdev->ht.height ||
  1060.     xdev->ht.pixmap == (Pixmap) 0) {
  1061.     if (xdev->ht.pixmap != (Pixmap) 0)
  1062.         XFreePixmap(xdev->dpy, xdev->ht.pixmap);
  1063.     xdev->ht.pixmap = XCreatePixmap(xdev->dpy, xdev->win,
  1064.                     tile->size.x, tile->size.y,
  1065.                     xdev->vinfo->depth);
  1066.     if (xdev->ht.pixmap == (Pixmap) 0)
  1067.         return 0;
  1068.     xdev->ht.width = tile->size.x, xdev->ht.height = tile->size.y;
  1069.     xdev->ht.raster = tile->raster;
  1070.     }
  1071.     xdev->ht.fore_c = xdev->fore_color;
  1072.     xdev->ht.back_c = xdev->back_color;
  1073.     /* Copy the tile into the Pixmap */
  1074.     xdev->image.data = (char *)tile->data;
  1075.     xdev->image.width = tile->size.x;
  1076.     xdev->image.height = tile->size.y;
  1077.     xdev->image.bytes_per_line = tile->raster;
  1078.     xdev->image.format = XYBitmap;
  1079.     set_fill_style(FillSolid);
  1080. #ifdef DEBUG
  1081.     if (gs_debug['H']) {
  1082.     int i;
  1083.  
  1084.     dprintf4("[H] 0x%lx: width=%d height=%d raster=%d\n",
  1085.          (ulong)tile->data, tile->size.x, tile->size.y, tile->raster);
  1086.     for (i = 0; i < tile->raster * tile->size.y; i++)
  1087.         dprintf1(" %02x", tile->data[i]);
  1088.     dputc('\n');
  1089.     }
  1090. #endif
  1091.     XSetTile(xdev->dpy, xdev->gc, xdev->ht.no_pixmap);    /* *** X bug *** */
  1092.     set_function(GXcopy);
  1093.     put_image(xdev->dpy, xdev->ht.pixmap, xdev->gc, &xdev->image,
  1094.           0, 0, 0, 0, tile->size.x, tile->size.y);
  1095.     XSetTile(xdev->dpy, xdev->gc, xdev->ht.pixmap);
  1096.     xdev->ht.id = tile->id;
  1097.     return xdev->useXSetTile;
  1098. }
  1099.  
  1100.  
  1101. /* ------ Screen update procedures ------ */
  1102.  
  1103. /* Flush updates to the screen if needed. */
  1104. private void
  1105. update_do_flush(register gx_device *dev)
  1106. {
  1107.     gx_device_X *xdev = (gx_device_X *)dev;
  1108.     int xo = xdev->update.xo, yo = xdev->update.yo;
  1109.  
  1110.     set_function(GXcopy);
  1111.     XCopyArea(xdev->dpy, xdev->bpixmap, xdev->win, xdev->gc,
  1112.           xo, yo, xdev->update.xe - xo, xdev->update.ye - yo,
  1113.           xo, yo);
  1114.     update_init(dev);
  1115. }
  1116.  
  1117. /* Add a region to be updated. */
  1118. /* This is only called if xdev->bpixmap != 0. */
  1119. void
  1120. x_update_add(gx_device *dev, int xo, int yo, int w, int h)
  1121. {
  1122.     register gx_device_X *xdev = (gx_device_X *)dev;
  1123.     int xe = xo + w, ye = yo + h;
  1124.     long new_area;
  1125.  
  1126.     if ( always_update )
  1127.       { /* Update the screen now. */
  1128.         update_do_flush(dev);
  1129.         new_area = (long)w * h;
  1130.       }
  1131.     else
  1132.       { /* Only update the screen if it's worthwhile. */
  1133.         if ( (++xdev->up_count >= 200 && xdev->up_area > 1000) ||
  1134.          xdev->up_area == 0
  1135.          )
  1136.           { if ( xdev->up_area != 0 )
  1137.           update_do_flush(dev);
  1138.         new_area = (long)w * h;
  1139.           }
  1140.         else
  1141.           { /* See whether adding this rectangle */
  1142.         /* would result in too much being copied unnecessarily. */
  1143.         long old_area = xdev->up_area;
  1144.         long new_up_area;
  1145.         rect u;
  1146.  
  1147.         u.xo = min(xo, xdev->update.xo);
  1148.         u.yo = min(yo, xdev->update.yo);
  1149.         u.xe = max(xe, xdev->update.xe);
  1150.         u.ye = max(ye, xdev->update.ye);
  1151.         new_up_area = (long)(u.xe - u.xo) * (u.ye - u.yo);
  1152.         /* The fraction of new_up_area used in the following test */
  1153.         /* is not particularly critical; using a denominator */
  1154.         /* that is a power of 2 eliminates a divide. */
  1155.         if ( u.xe - u.xo >= 10 && u.ye - u.yo >= 10 &&
  1156.              old_area + (new_area = (long)w * h) <
  1157.               new_up_area - (new_up_area >> 2)
  1158.            )
  1159.           update_do_flush(dev);
  1160.         else
  1161.           {    xdev->update = u;
  1162.             xdev->up_area = new_up_area;
  1163.             return;
  1164.           }
  1165.           }
  1166.       }
  1167.  
  1168.     xdev->update.xo = xo;
  1169.     xdev->update.yo = yo;
  1170.     xdev->update.xe = xe;
  1171.     xdev->update.ye = ye;
  1172.     xdev->up_area = new_area;
  1173. }
  1174.  
  1175. /* ------ Internal procedures ------ */
  1176.  
  1177. /* Substitute for XPutImage using XFillRectangle. */
  1178. /* This is a total hack to get around an apparent bug */
  1179. /* in some X servers.  It only works with the specific */
  1180. /* parameters (bit/byte order, padding) used above. */
  1181. private int
  1182. alt_put_image(gx_device *dev, Display *dpy, Drawable win, GC gc,
  1183.       XImage *pi, int sx, int sy, int dx, int dy, unsigned w, unsigned h)
  1184. {
  1185.     int raster = pi->bytes_per_line;
  1186.     byte *data = (byte *) pi->data + sy * raster + (sx >> 3);
  1187.     int init_mask = 0x80 >> (sx & 7);
  1188.     int invert = 0;
  1189.     int yi;
  1190.  
  1191. #define nrects 40
  1192.     XRectangle rects[nrects];
  1193.     XRectangle *rp = rects;
  1194.  
  1195.     XGCValues gcv;
  1196.  
  1197.     XGetGCValues(dpy, gc, (GCFunction | GCForeground | GCBackground), &gcv);
  1198.  
  1199.     if (gcv.function == GXcopy) {
  1200.     XSetForeground(dpy, gc, gcv.background);
  1201.     XFillRectangle(dpy, win, gc, dx, dy, w, h);
  1202.     XSetForeground(dpy, gc, gcv.foreground);
  1203.     } else if (gcv.function == GXand) {
  1204.     if (gcv.background != ~(x_pixel) 0) {
  1205.         XSetForeground(dpy, gc, gcv.background);
  1206.         invert = 0xff;
  1207.     }
  1208.     } else if (gcv.function == GXor) {
  1209.     if (gcv.background != 0) {
  1210.         XSetForeground(dpy, gc, gcv.background);
  1211.         invert = 0xff;
  1212.     }
  1213.     } else {
  1214.     lprintf("alt_put_image: unimplemented function.\n");
  1215.     return_error(gs_error_rangecheck);
  1216.     }
  1217.  
  1218.     for (yi = 0; yi < h; yi++, data += raster) {
  1219.     register int mask = init_mask;
  1220.     register byte *dp = data;
  1221.     register int xi = 0;
  1222.  
  1223.     while (xi < w) {
  1224.         if ((*dp ^ invert) & mask) {
  1225.         int xleft = xi;
  1226.  
  1227.         if (rp == &rects[nrects]) {
  1228.           XFillRectangles(dpy, win, gc, rects, nrects);
  1229.           rp = rects;
  1230.         }
  1231.         /* Scan over a run of 1-bits */
  1232.         rp->x = dx + xi, rp->y = dy + yi;
  1233.         do {
  1234.             if (!(mask >>= 1))
  1235.             mask = 0x80, dp++;
  1236.             xi++;
  1237.         } while (xi < w && (*dp & mask));
  1238.         rp->width = xi - xleft, rp->height = 1;
  1239.         rp++;
  1240.         } else {
  1241.         if (!(mask >>= 1))
  1242.             mask = 0x80, dp++;
  1243.         xi++;
  1244.         }
  1245.     }
  1246.     }
  1247.     XFillRectangles(dpy, win, gc, rects, rp - rects);
  1248.     if (invert) XSetForeground(dpy, gc, gcv.foreground);
  1249.     return 0;
  1250. }
  1251.